linux下system函数调用shell命令后,怎样让主进程等子进程返回后,接着执行 | 您所在的位置:网站首页 › shell 调用函数 › linux下system函数调用shell命令后,怎样让主进程等子进程返回后,接着执行 |
这是进程间同步的问题。解决方法是:fork一个子进程执行system调用,父进程调用 wait 或 waitpid 等待子进程的终止信息。
父进程调用 wait 或 waitpid 时可能会: • 阻塞(如果它的所有子进程都还在运行)。 • 带子进程的终止信息立即返回(如果一个子进程已终止,正等待父进程读取其终止信息)。 • 出错立即返回(如果它没有任何子进程)。
wait 和 waitpid 这两个函数的区别是: • 如果父进程的所有子进程都还在运行,调用wait将使父进程阻塞,而调用waitpid时如果在options参数中指定WNOHANG可以使父进程不阻塞而立即返回0。 • wait等待第一个终止的子进程,而waitpid可以通过pid参数指定等待哪一个子进程。 Linux中启动另一个可执行文件或程序用system函数最理想了,这个函数将在你编写的那个程序的内部启动另一个程序,从而创建一个新进程,并等待这个进程执行完毕退出。如果正常执行,system函数将返回被执行程序的退出码;如果无法运行这个程序,将返回错误代码127;如果是其他错误,返回-1。这个函数的原型是:#include stdlib.h int system(const char *string) 参数string是将要执行的程序的命令字符串。 还有一种执行外部程序的方法是exec系列函数,但这个系列的函数都是将当前进程的替换成新进程,也就是说原来的进程不存在了。 linux c system函数介绍:
system(执行shell 命令) 相关函数 fork,execve,waitpid,popen 表头文件 #i nclude 定义函数 int system(const char * string) 函数说明 system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。 返回值 =-1:出现错误 =0:调用成功但是没有出现子进程 0:成功退出的子进程的id 如果system()在调用/bin/sh时失败则返回127,其他失败原因返回-1。若参数string为空指针(NULL),则返回非零值。如果system()调用成功则最后会返回执行shell命令后的返回值,但是此返回值也有可能为 system()调用/bin/sh失败所返回的127,因此最好能再检查errno 来确认执行成功。 附加说明 在编写具有SUID/SGID权限的程序时请勿使用system(),system()会继承环境变量,通过环境变量可能会造成系统安全的问题。 范例 #i nclude main() { system("ls -al /etc/passwd /etc/shadow") } 执行结果: -rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd -r--------- 1 root root 572 Sep 2 15 :34 /etc/shado 例2: char tmp[] sprintf(tmp,"/bin/mount -t vfat %s /mnt/usb",dev) system(tmp) 其中dev是/dev/sda1。 system函数的源码 #include syspes.h #include sys/wait.h #include errno.h #include unistd.h int system(const char * cmdstring) { pid_t pid int status if(cmdstring == NULL){ return (1) } if((pid = fork())0){ status = -1 } else if(pid = 0){ execl("/bin/sh", "sh", "-c", cmdstring, (char *)0) -exit(127)//子进程正常执行则不会执行此语句 } else { while(waitpid(pid, status, 0) 0){ if(errno != EINTER) { status = -1 break } } } return status } 那么如何获得system的返回值呢?? char buf[10] char * ps="ps -ef|grep -c root" FILE *ptr int i if((ptr = popen(ps, "r")) != NULL) { fgets(buf, 10 , ptr) i = atoi(buf) pclose(ptr) } 可以man下waitpid查看下如何检查status的值 int ret = system("ls -al /etc/passwd /etc/shadow") if(WIFSIGNALED(ret)) 具体的这些宏查看man waitpid 欢迎分享,转载请注明来源:内存溢出 原文地址:https://outofmemory.cn/yw/8395107.html |
CopyRight 2018-2019 实验室设备网 版权所有 |